/**
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.activemq.transport.tcp;

import javax.net.SocketFactory;
import java.io.IOException;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Random;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Automatically generated socket.close() calls to simulate network faults
 */
public class SocketTstFactory extends SocketFactory {

   private static final Logger LOG = LoggerFactory.getLogger(SocketTstFactory.class);

   private static final ConcurrentMap<InetAddress, Integer> closeIter = new ConcurrentHashMap<>();

   private class SocketTst {

      private class Bagot implements Runnable {

         private final Thread processus;
         private final Socket socket;
         private final InetAddress address;

         public Bagot(Random rnd, Socket socket, InetAddress address) {
            this.processus = new Thread(this, "Network Faults maker : undefined");
            this.socket = socket;
            this.address = address;
         }

         public void start() {
            this.processus.setName("Network Faults maker : " + this.socket.toString());
            this.processus.start();
         }

         @Override
         public void run() {
            int lastDelayVal;
            Integer lastDelay;
            while (!this.processus.isInterrupted()) {
               if (!this.socket.isClosed()) {
                  try {
                     lastDelay = closeIter.get(this.address);
                     if (lastDelay == null) {
                        lastDelayVal = 0;
                     } else {
                        lastDelayVal = lastDelay.intValue();
                        if (lastDelayVal > 10)
                           lastDelayVal += 20;
                        else
                           lastDelayVal += 1;
                     }

                     lastDelay = new Integer(lastDelayVal);

                     LOG.info("Trying to close client socket " + socket.toString() + " in " + lastDelayVal + " milliseconds");

                     try {
                        Thread.sleep(lastDelayVal);
                     } catch (InterruptedException e) {
                        this.processus.interrupt();
                        Thread.currentThread().interrupt();
                     } catch (IllegalArgumentException e) {
                     }

                     this.socket.close();
                     closeIter.put(this.address, lastDelay);
                     LOG.info("Client socket " + this.socket.toString() + " is closed.");
                  } catch (IOException e) {
                  }
               }

               this.processus.interrupt();
            }
         }
      }

      private final Bagot bagot;
      private final Socket socket;

      public SocketTst(InetAddress address, int port, Random rnd) throws IOException {
         this.socket = new Socket(address, port);
         bagot = new Bagot(rnd, this.socket, address);
      }

      public SocketTst(InetAddress address,
                       int port,
                       InetAddress localAddr,
                       int localPort,
                       Random rnd) throws IOException {
         this.socket = new Socket(address, port, localAddr, localPort);
         bagot = new Bagot(rnd, this.socket, address);
      }

      public SocketTst(String address, int port, Random rnd) throws UnknownHostException, IOException {
         this.socket = new Socket(address, port);
         bagot = new Bagot(rnd, this.socket, InetAddress.getByName(address));
      }

      public SocketTst(String address, int port, InetAddress localAddr, int localPort, Random rnd) throws IOException {
         this.socket = new Socket(address, port, localAddr, localPort);
         bagot = new Bagot(rnd, this.socket, InetAddress.getByName(address));
      }

      public Socket getSocket() {
         return this.socket;
      }

      public void startBagot() {
         bagot.start();
      }
   }

   private final Random rnd;

   public SocketTstFactory() {
      super();
      LOG.info("Creating a new SocketTstFactory");
      this.rnd = new Random();
   }

   @Override
   public Socket createSocket(InetAddress host, int port) throws IOException {
      SocketTst sockTst;
      sockTst = new SocketTst(host, port, this.rnd);
      sockTst.startBagot();
      return sockTst.getSocket();
   }

   @Override
   public Socket createSocket(InetAddress host, int port, InetAddress localAddress, int localPort) throws IOException {
      SocketTst sockTst;
      sockTst = new SocketTst(host, port, localAddress, localPort, this.rnd);
      sockTst.startBagot();
      return sockTst.getSocket();
   }

   @Override
   public Socket createSocket(String host, int port) throws IOException {
      SocketTst sockTst;
      sockTst = new SocketTst(host, port, this.rnd);
      sockTst.startBagot();
      return sockTst.getSocket();
   }

   @Override
   public Socket createSocket(String host, int port, InetAddress localAddress, int localPort) throws IOException {
      SocketTst sockTst;
      sockTst = new SocketTst(host, port, localAddress, localPort, this.rnd);
      sockTst.startBagot();
      return sockTst.getSocket();
   }

   private final static SocketTstFactory client = new SocketTstFactory();

   public static SocketFactory getDefault() {
      return client;
   }
}
